<template>
  <div class='relative' ref='containerTarget' :style='{height: containerHeight + "px"}'>
    <!-- 数据渲染 -->
    <template v-if='columnWidth && data.length'>
      <div
        v-for='(item,index) in data'
        :key='nodeKey? item[nodeKey]:index'
        class='m-waterfall-item absolute duration-300'
        :style='{
          width:columnWidth + "px",
          left: item._style?.left + "px",
          top: item._style?.top + "px"
        }'
      >
        <slot :item='item' :width='columnWidth' :index='index' />
      </div>
    </template>
    <!-- 加载中 -->
    <div v-else>加载中...</div>
  </div>
</template>

<script setup>
import { computed, nextTick, onMounted, onUnmounted, ref, watch } from 'vue'
import {
  getAllImg,
  getImgElements,
  getMaxHeight,
  getMinHeight,
  getMinHeightColumn,
  onCompleteImg
} from '@/libs/water-fall/util'

const props = defineProps({
  // 数据源
  data: {
    type: Array,
    required: true
  },
  // 唯一标识
  nodeKey: {
    type: String
  },
  // 列数
  column: {
    default: 2,
    type: Number
  },
  // 列间距
  columnSpacing: {
    default: 20,
    type: Number
  },
  // 行间距
  rowSpacing: {
    default: 20,
    type: Number
  },
  // 是否需要进行图片预读取
  picturePreReading: {
    type: Boolean,
    default: true
  }
})

// 容器的总高度
const containerHeight = ref(0)
// 记录每一列高度的容器
const columnHeightObj = ref({})
/**
 * 构建记录每列高度的对象
 */
const useColumnHeightObj = () => {
  columnHeightObj.value = {}
  for (let i = 0; i < props.column; i++) {
    columnHeightObj.value[i] = 0
  }
}

// 容器的宽度
const containerTarget = ref(null)
const containerWidth = ref(0)
const containerLeft = ref(0)
const useContainerWidth = () => {
  const { paddingLeft, paddingRight } = getComputedStyle(containerTarget.value, null)
  // 容器左边距
  containerLeft.value = parseFloat(paddingLeft)
  // 容器的宽度
  containerWidth.value = containerTarget.value.offsetWidth - parseFloat(paddingLeft) - parseFloat(paddingRight)
}

// 列宽
const columnWidth = ref(0)
// 列间距的合计
const columnSpacingTotal = computed(() => {
  return (props.column - 1) * props.columnSpacing
})

// 计算列宽
const useColumnWidth = () => {
  // 计算容器宽邸
  useContainerWidth()
  // 计算列宽
  columnWidth.value = (containerWidth.value - columnSpacingTotal.value) / props.column
}

onMounted(() => {
  // 计算列宽
  useColumnWidth()
})
// item 高度集合
let itemHeights = []
/**
 * 监听图片加载完成 (需要图片预加载）
 */
const waitImgComplete = () => {
  itemHeights = []
  let itemElements = [...document.getElementsByClassName('m-waterfall-item')]
  // 获取到元素的 img 标签
  const imgElements = getImgElements(itemElements)
  // 获取所有 img 标签的图片
  const allImg = getAllImg(imgElements)
  // 等待图片加载完成
  onCompleteImg(allImg).then(() => {
    // 图片加载完成
    itemElements.forEach(el => {
      itemHeights.push(el.offsetHeight)
    })
    // 渲染位置
    useItemLocation()
  })
}
/**
 * 不需要图片预加载
 */
const useItemHeight = () => {
  itemHeights = []
  // 拿到所有的元素
  let itemElements = [...document.getElementsByClassName('m-waterfall-item')]
  // 计算 item 高度
  itemElements.forEach(el => {
    itemHeights.push(el.offsetHeight)
  })
  useItemLocation()
}

/**
 * 渲染位置
 */
const useItemLocation = () => {
  props.data.forEach((item,index) => {
    // 避免重复计算
    if (item._style) {
      return
    }
    item._style = {}
    item._style.left = getItemLeft()
    item._style.top = getItemTop()
    // 指定的列高度的自增
    increaseHeight(index)
  })
  // 指定容器的高度
  containerHeight.value = getMaxHeight(columnHeightObj.value)
}

/**
 * 在组件销毁的时候，清除所有的 _style
 */
onUnmounted(() => {
  props.data.forEach(item => {
    delete item._style
  })
})
/**
 * 返回下一个 item 的 left
 */
const getItemLeft = () => {
  const column = getMinHeightColumn(columnHeightObj.value)
  return column * (columnWidth.value + props.columnSpacing) + containerLeft.value
}
/**
 * 返回下一个 item 的 top
 */
const getItemTop = () => {
  return getMinHeight(columnHeightObj.value)
}
const increaseHeight = (index) => {
  // 拿到最小高度所在的列
  const minHeightColumn = getMinHeightColumn(columnHeightObj.value)
  // 使用该列自增
  columnHeightObj.value[minHeightColumn] += itemHeights[index] + props.rowSpacing
}
/**
 * 触发计算
 */
watch(
  () => props.data,
  (newVal) => {
    nextTick(() => {
      // 第一次获取数据时，构建高度记录容器
      const resetColumnHeight = newVal.every((item) => !item._style)
      if (resetColumnHeight) {
        useColumnHeightObj()
      }
      if (props.picturePreReading) {
        waitImgComplete()
      } else {
        useItemHeight()
      }
    })
  },
  {
    deep:true,
    immediate: true
  }
)

/**
 * 重新构建瀑布流
 */
const reset = () => {
  setTimeout(()=>{
    useColumnWidth()
    props.data.forEach(item=> {
      item._style = null
    })
  },100)
}
/**
 * 监听列数的变化
 */
watch(
  ()=> props.column,
  () => {
    if (props.picturePreReading) {
      columnWidth.value = 0
      /**
       * 数据改变之后，视图改变之后的回调
       */
      reset()
    }else {
      reset()
    }
  }
)
</script>

<style lang='scss' scoped>
</style>
